﻿using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace ConsoleApp
{
    /*
     * 罗马数字包含以下七种字符: I， V， X， L，C，D 和 M。
    字符          数值
    I             1
    V             5
    X             10
    L             50
    C             100
    D             500
    M             1000
    例如， 罗马数字 2 写做 II ，即为两个并列的 1。12 写做 XII ，即为 X + II 。 27 写做  XXVII, 即为 XX + V + II 。
    通常情况下，罗马数字中小的数字在大的数字的右边。但也存在特例，例如 4 不写做 IIII，而是 IV。数字 1 在数字 5 的左边，所表示的数等于大数 5 减小数 1 得到的数值 4 。同样地，数字 9 表示为 IX。这个特殊的规则只适用于以下六种情况：
    I 可以放在 V (5) 和 X (10) 的左边，来表示 4 和 9。
    X 可以放在 L (50) 和 C (100) 的左边，来表示 40 和 90。 
    C 可以放在 D (500) 和 M (1000) 的左边，来表示 400 和 900。
    给定一个罗马数字，将其转换成整数。输入确保在 1 到 3999 的范围内。
    示例 1:
    输入: "III"
    输出: 3
    示例 2:
    输入: "IV"
    输出: 4
    示例 3:
    输入: "IX"
    输出: 9
    示例 4:
    输入: "LVIII"
    输出: 58
    解释: L = 50, V= 5, III = 3.
    示例 5:
    输入: "MCMXCIV"
    输出: 1994
    解释: M = 1000, CM = 900, XC = 90, IV = 4.    
     */
    class Action013_罗马数字转换整数
    {
        public void Test()
        {
            Console.WriteLine("III Expect=3; Actual=" + RomanToInt("III"));
            Console.WriteLine("IV Expect=4; Actual=" + RomanToInt("IV"));
            Console.WriteLine("IX Expect=9; Actual=" + RomanToInt("IX"));
            Console.WriteLine("LVIII Expect=58; Actual=" + RomanToInt("LVIII"));
            Console.WriteLine("MCMXCIV Expect=1994; Actual=" + RomanToInt("MCMXCIV"));
        }

        public int RomanToInt(string s)
        {
            //数值转换表
            Dictionary<string, int> priorities = new Dictionary<string, int>()
            {
                { "IV", 4},
                { "IX", 9},
                { "XL", 40},
                { "XC", 90},
                { "CD", 400},
                { "CM", 900},
                { "I", 1},
                { "V", 5},
                { "X", 10},
                { "L", 50},
                { "C", 100},
                { "D", 500},
                { "M", 1000},
            };

            int value = 0;
            while (s != string.Empty)
            {
                if (s.Length >= 2)
                {
                    //如果剩余长度超过2，先测试前两个字符是否是特殊值（4，9...）
                    string pop2 = s.Substring(0, 2);
                    if (priorities.TryGetValue(pop2, out int val))
                    {
                        //如果是特殊值，则此处做操作，并跳入下一循环
                        value += val;
                        s = s.Substring(2);
                        continue;
                    }
                }

                //如果不是特殊值，则此处操作，并跳入下一循环
                string pop = s.Substring(0, 1);
                value += priorities[pop];
                s = s.Substring(1);
            }

            return value;
        }
    }
}
